文章目录
  1. 1. 构造函数
  2. 2. 链表的添加
  3. 3. 链表的删除
  4. 4. 链表的get
  5. 5. Iterator方法
  6. 6. 一些便利的方法
  7. 7. 总结

LinkedListJava编程中最常用的容器类之一,它是基于双向链接实现的,在面试时常常被与ArrayList作比较!

       LinkedList容器在插入操作上效率很高,本文主要以他的构造函数,存储结构,增删改机制来介绍LinkedList

构造函数

       先来看一下它的几个私有变量的定义:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
transient int size = 0;

/**
* Pointer to first node.
* Invariant: (first == null && last == null) ||
* (first.prev == null && first.item != null)
*/

transient Node<E> first;

/**
* Pointer to last node.
* Invariant: (first == null && last == null) ||
* (last.next == null && last.item != null)
*/

transient Node<E> last;

private static class Node<E> {
E item;
Node<E> next;
Node<E> prev;

Node(Node<E> prev, E element, Node<E> next) {
this.item = element;
this.next = next;
this.prev = prev;
}
}

       注释中描述的得清晰

  • size表示LinkedList列表的实际尺寸大小
  • firstlast分别表示双向链表的头部指针和尾部指针,默认都是指向null
  • NodeLinkedList中一个内部类,单个元素存储的实体,通过prevnext来建立双向链表

       咱们在来看一下LinkedList的构造函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
* Constructs an empty list.
*/

public LinkedList() {
}

/**
* Constructs a list containing the elements of the specified
* collection, in the order they are returned by the collection's
* iterator.
*
* @param c the collection whose elements are to be placed into this list
* @throws NullPointerException if the specified collection is null
*/

public LinkedList(Collection<? extends E> c) {
this();
addAll(c);
}

       LinkedList在它的构造函数上有一个无参构造器和集合类构造器,这个大致和ArrayList是类似的,少了一个指定初始容量构造器。

链表的添加

       先来看下源码中链表添加元素的几个基础实现方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
/**
* Links e as first element.添加一个元素作为链表的头部
*/

private void linkFirst(E e) {
final Node<E> f = first;
final Node<E> newNode = new Node<>(null, e, f);
first = newNode;
if (f == null)
last = newNode;//如果链表里面还没有元素 则将第一个元素置为last 当然他同时也是first
else
f.prev = newNode;
size++;
modCount++;
}

/**
* Links e as last element.//添加一个元素作为链表的尾部
*/

void linkLast(E e) {
final Node<E> l = last;
final Node<E> newNode = new Node<>(l, e, null);
last = newNode;
if (l == null)
first = newNode;//如果链表里面没有元素,则这个添加的元素为first,同时也是last
else
l.next = newNode;
size++;
modCount++;
}

/**
* Inserts element e before non-null Node succ.//在非空的succ之前插入元素
*/

void linkBefore(E e, Node<E> succ) {
// assert succ != null;
final Node<E> pred = succ.prev;
final Node<E> newNode = new Node<>(pred, e, succ);
succ.prev = newNode;
if (pred == null)
first = newNode;
else
pred.next = newNode;
size++;
modCount++;
}

       在上面的添加功能的基础实现方法中,linkFirstlinkLast这两个方法很容易理解,它们是通过改变头部指针或者尾部指针的指向来实现将元素直接添加到头部或者尾部,linkBefore这个方法这则需要指定一个插入元素的“坑”, 再往这个“坑”前进行元素添加,用图来表示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
/**
* Inserts the specified element at the beginning of this list.
*
* @param e the element to add
*/

public void addFirst(E e) {
linkFirst(e);
}

/**
* Appends the specified element to the end of this list.
*
* <p>This method is equivalent to {@link #add}.
*
* @param e the element to add
*/

public void addLast(E e) {
linkLast(e);
}
/**
* Appends the specified element to the end of this list.
*
* <p>This method is equivalent to {@link #addLast}.
*
* @param e element to be appended to this list
* @return {@code true} (as specified by {@link Collection#add})
*/

public boolean add(E e) {
linkLast(e);//默认插入的元素都是插入到链表的尾部
return true;
}
/**
* Inserts the specified element at the specified position in this list.
* Shifts the element currently at that position (if any) and any
* subsequent elements to the right (adds one to their indices).
*
* @param index index at which the specified element is to be inserted
* @param element element to be inserted
* @throws IndexOutOfBoundsException {@inheritDoc}
*/

public void add(int index, E element) {
checkPositionIndex(index);

if (index == size)
linkLast(element);
else
linkBefore(element, node(index));

}

       上面源码中的方法均是LinkedList公开出来的方法,最终也都是调用了linkFirst,linkLast,linkBefore来完成操作的。

链表的删除

       下面贴出来的是链表删除操作时的核心方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
/**
* Unlinks non-null first node f.//
*/

private E unlinkFirst(Node<E> f) {//将fisrt指针指向f.next 也就是删除了f节点以及他之前的元素
// assert f == first && f != null;
final E element = f.item;
final Node<E> next = f.next;
f.item = null;
f.next = null; // help GC
first = next;
if (next == null)
last = null;
else
next.prev = null;
size--;
modCount++;
return element;
}

/**
* Unlinks non-null last node l.//
*/

private E unlinkLast(Node<E> l) {//将last指针指针l.prev,也就是删除了f节点以及他之后的元素
// assert l == last && l != null;
final E element = l.item;
final Node<E> prev = l.prev;
l.item = null;
l.prev = null; // help GC
last = prev;
if (prev == null)
first = null;
else
prev.next = null;
size--;
modCount++;
return element;
}

/**
* Unlinks non-null node x.
*/

E unlink(Node<E> x) {//删除指定的元素x
// assert x != null;
final E element = x.item;
final Node<E> next = x.next;
final Node<E> prev = x.prev;

if (prev == null) {
first = next;
} else {
prev.next = next;
x.prev = null;
}

if (next == null) {
last = prev;
} else {
next.prev = prev;
x.next = null;
}

x.item = null;
size--;
modCount++;
return element;
}

       删除方法中unlinkFirst和unlinkLast一眼看上去比较难理解,不过稍微盯几分钟画个图还是很好理解的

  • unlinkFirst(Node f):将fisrt指针指向f.next 也就是删除了f节点以及他之前的元素
  • unlinkLast(Node l):将last指针指针l.prev,也就是删除了l节点以及他之后的元素
  • unlink(Node x):最容易理解,直接删除指定的x元素
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
/**
* Removes and returns the first element from this list.
*
* @return the first element from this list
* @throws NoSuchElementException if this list is empty
*/

public E removeFirst() {
final Node<E> f = first;
if (f == null)
throw new NoSuchElementException();
return unlinkFirst(f);
}

/**
* Removes and returns the last element from this list.
*
* @return the last element from this list
* @throws NoSuchElementException if this list is empty
*/

public E removeLast() {
final Node<E> l = last;
if (l == null)
throw new NoSuchElementException();
return unlinkLast(l);
}

/**
* Removes the first occurrence of the specified element from this list,
* if it is present. If this list does not contain the element, it is
* unchanged. More formally, removes the element with the lowest index
* {@code i} such that
* <tt>(o==null&nbsp;?&nbsp;get(i)==null&nbsp;:&nbsp;o.equals(get(i)))</tt>
* (if such an element exists). Returns {@code true} if this list
* contained the specified element (or equivalently, if this list
* changed as a result of the call).
*
* @param o element to be removed from this list, if present
* @return {@code true} if this list contained the specified element
*/

public boolean remove(Object o) {
if (o == null) {
for (Node<E> x = first; x != null; x = x.next) {
if (x.item == null) {
unlink(x);
return true;
}
}
} else {
for (Node<E> x = first; x != null; x = x.next) {
if (o.equals(x.item)) {
unlink(x);
return true;
}
}
}
return false;
}

       可以看到LinkedList删除里面的公开方法也都是直接调用了unlinkFirst,unlinkLast,remove

链表的get

LinkedList是基于链表实现的,其插入和删除元素方法都是在O(1)里面完成的,那么他的get方法又是如何做的呢?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/**
* Returns the (non-null) Node at the specified element index.
*/

Node<E> node(int index) {
// assert isElementIndex(index);

if (index < (size >> 1)) {
Node<E> x = first;
for (int i = 0; i < index; i++)//从前往后取
x = x.next;
return x;
} else {
Node<E> x = last;
for (int i = size - 1; i > index; i--)//从后往前取
x = x.prev;
return x;
}
}

/**
* Returns the element at the specified position in this list.
*
* @param index index of the element to return
* @return the element at the specified position in this list
* @throws IndexOutOfBoundsException {@inheritDoc}
*/

public E get(int index) {
checkElementIndex(index);
return node(index).item;
}

       从源码的可以看到,LinkedList的get方法不能像ArrayList一样直接按索引取值,这里需要用遍历的方法来做,那当然在get上的复杂度肯定是要高于ArrayList了,不过LinkedList也做了一定的优化,从node(int index)可以看到,在按索引遍历的时候会根据索引所在位置(前半部分,后半部分)来决定遍历的方向:

  • 当index<size/2的时候,从前往后取
  • 当index>size/2的时候,从后往前取

Iterator方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
private class ListItr implements ListIterator<E> {
private Node<E> lastReturned = null;
private Node<E> next;
private int nextIndex;
private int expectedModCount = modCount;

ListItr(int index) {
// assert isPositionIndex(index);
next = (index == size) ? null : node(index);
nextIndex = index;
}

public boolean hasNext() {
return nextIndex < size;
}

public E next() {
checkForComodification();
if (!hasNext())
throw new NoSuchElementException();

lastReturned = next;
next = next.next;
nextIndex++;
return lastReturned.item;
}

public boolean hasPrevious() {
return nextIndex > 0;
}

public E previous() {
checkForComodification();
if (!hasPrevious())
throw new NoSuchElementException();

lastReturned = next = (next == null) ? last : next.prev;
nextIndex--;
return lastReturned.item;
}

public int nextIndex() {
return nextIndex;
}

public int previousIndex() {
return nextIndex - 1;
}

public void remove() {
checkForComodification();
if (lastReturned == null)
throw new IllegalStateException();

Node<E> lastNext = lastReturned.next;
unlink(lastReturned);
if (next == lastReturned)
next = lastNext;
else
nextIndex--;
lastReturned = null;
expectedModCount++;
}

public void set(E e) {
if (lastReturned == null)
throw new IllegalStateException();
checkForComodification();
lastReturned.item = e;
}

public void add(E e) {
checkForComodification();
lastReturned = null;
if (next == null)
linkLast(e);
else
linkBefore(e, next);

nextIndex++;
expectedModCount++;
}

final void checkForComodification() {
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
}
}

       在ArrayListiterator方法里面只提供了向后遍历,但是在LinkedListiterator中可以发现他在提供hasNext的同时还提供了hasPrevious方法,所以这个iterator双向遍历功能,既可以从前往后,也可以从后往前进行遍历,为我们在开发时提供了不少的便利

一些便利的方法

  • peek(),peekFirst()返回链表的头部但不删除,poll(),pollFirst()返回链表的头部同时将其删除
  • peekLast()返回链表的尾部但是不删除,pollLast()返回链表的尾部同时将其删除
  • offer(E e),offerLast(E e)在链表的尾部添加一个元素,offerFirst(E e)向链表的头部添加一个元素
  • removeFirstOccurrence(Object o)移除第一次出现的元素,removeLastOccurrence(Object o)移除最后一次出现的元素

总结

  • LinkedList是基于双向链表实现的
  • LinkedList的插入和删除的复杂度为O(1),(假设已有插入或者删除的元素),get的复杂度为O(N)
  • LinkedList的迭代器有双向遍历功能

本作品采用[知识共享署名-非商业性使用-相同方式共享 2.5]中国大陆许可协议进行许可,我的博客欢迎复制共享,但在同时,希望保留我的署名权kubiCode,并且,不得用于商业用途。如您有任何疑问或者授权方面的协商,请给我留言

文章目录
  1. 1. 构造函数
  2. 2. 链表的添加
  3. 3. 链表的删除
  4. 4. 链表的get
  5. 5. Iterator方法
  6. 6. 一些便利的方法
  7. 7. 总结